import random
import re

import datetime
from flask import current_app, jsonify
from flask import make_response
from flask import request
from flask import session
from flask.ext.sqlalchemy import SQLAlchemy

from info import constants, db
from info import redis_store
from info.libs.yuntongxun.sms import CCP
from info.models import User
from info.utils.captcha.captcha import captcha
from info.utils.response_code import RET
from . import passport_blu

# 发送短信
@passport_blu.route('/smscode',methods=['post'])
def send_sms():
    """
    1. 接收参数并判断是否有值
    2. 校验手机号是正确
    3. 通过传入的图片编码去redis中查询真实的图片验证码内容
    4. 进行验证码内容的比对
    5. 生成发送短信的内容并发送短信
    6. redis中保存短信验证码内容
    7. 返回发送成功的响应
    :return:
    """
    # 1. 接收参数并判断是否有值
    # 取到请求值中的内容
    args_data = request.json
    mobile = args_data.get("mobile")
    image_code = args_data.get("image_code")
    image_code_id = args_data.get("image_code_id")

    if not all([mobile, image_code_id, image_code]):
        # 参数不全
        return jsonify(errno=RET.PARAMERR, errmsg="参数不全")

    # 2. 校验手机号是正确
    if not re.match("^1[3578][0-9]{9}$", mobile):
        # 提示手机号不正确
        return jsonify(errno=RET.DATAERR, errmsg="手机号不正确")

    # 3. 通过传入的图片编码去redis中查询真实的图片验证码内容
    try:
        real_image_code = redis_store.get("ImageCode_" + image_code_id)
        # 如果能够取出来值，删除redis中缓存的内容
        if real_image_code:
            real_image_code = real_image_code.decode()
            redis_store.delete("ImageCode_" + image_code_id)
    except Exception as e:
        current_app.logger.error(e)
        # 获取图片验证码失败
        return jsonify(errno=RET.DBERR, errmsg="获取图片验证码失败")
    # 3.1 判断验证码是否存在，已过期
    if not real_image_code:
        # 验证码已过期
        return jsonify(errno=RET.NODATA, errmsg="验证码已过期")

    # 4. 进行验证码内容的比对
    if image_code.lower() != real_image_code.lower():
        # 验证码输入错误
        return jsonify(errno=RET.DATAERR, errmsg="验证码输入错误")

    # 4.1 校验该手机是否已经注册
    try:
        user = User.query.filter_by(mobile=mobile).first()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="数据库查询错误")
    if user:
        # 该手机已被注册
        return jsonify(errno=RET.DATAEXIST, errmsg="该手机已被注册")

    # 5. 生成发送短信的内容并发送短信
    result = random.randint(0, 999999)
    sms_code = "%06d" % result
    current_app.logger.debug("短信验证码的内容：%s" % sms_code)
    result = CCP().send_template_sms(mobile, [sms_code, constants.SMS_CODE_REDIS_EXPIRES / 60], "1")
    if result != 0:
        # 发送短信失败
        return jsonify(errno=RET.THIRDERR, errmsg="发送短信失败")

    # 6. redis中保存短信验证码内容
    try:
        redis_store.set("SMS_" + mobile, sms_code, constants.SMS_CODE_REDIS_EXPIRES)
    except Exception as e:
        current_app.logger.error(e)
        # 保存短信验证码失败
        return jsonify(errno=RET.DBERR, errmsg="保存短信验证码失败")

    # 7. 返回发送成功的响应
    return jsonify(errno=RET.OK, errmsg="发送成功")


"""
  1. 获取参数
  2. 生成验证码
  3. 删除之前验证码并保存当前验证码
  4. 错误处理
  5. 响应返回
  """
# 返回图片验证码
@passport_blu.route('/image_code')
def get_img_code():
     # 获取当前的图片编号id
    code_id = request.args.get('code_id')
    # 生成验证码
    name, text, image = captcha.generate_captcha()
    try:
        # 保存当前生成的图片验证码内容
        redis_store.setex('ImageCode_'+code_id,constants.IMAGE_CODE_REDIS_EXPIRES,text)
    except Exception as e:
        current_app.logger.error(e)
        return make_response(jsonify(errno=RET.DATAERR,errmsg='保存图片验证码失败'))

    # 返回响应内容
    resp = make_response(image)

    # 设置内容类型
    resp.headers['Content-Type'] = 'image/jpg'
    return resp

# 注册
@passport_blu.route('/register', methods=["POST"])
def register():
    """
    自己分析
    1. 获取前端参数
    2. 判断参数是否获取完全
    3. 验证手机号码格式是否正确
    4. 查看数据库中是否已有手机号码
    5. 验证验证码是否正确
    6. 将数据保存在数据库当中
    7. 将数据存在session中，表示已经登录
    :return:
    """
    # 1. 获取参数
    json_data = request.json
    mobile = json_data.get("mobile")
    password = json_data.get("password")
    sms_code = json_data.get("smscode")

    # 2. 判断参数
    if not all([mobile,password,sms_code]):
        return jsonify(errno=RET.PARAMERR, errmsg="参数不完整")

    # 3. 判断手机号码格式
    if not re.match("^1[345678]//d{9}",mobile):
        return jsonify(errno=RET.DATAERR, errmsg = "手机号格式错误")

    #
    # TODO 4. 查询数据库中是否存在手机号码

    # 5. 判断验证码是否正确
    try:
        rel_sms_code = redis_store.get("SMS_"+mobile)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="数据库查询错误")

    if not rel_sms_code:
        return jsonify(errno= RET.DBERR, errmsg = "数据已经过期")

    if rel_sms_code != sms_code:
        return jsonify(errno = RET.DATAERR, errmsg = "短信验证码错误")

    # 删除验证码
    try:
        redis_store.delete('SMS_'+mobile)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno = RET.DBERR, errmsg ="删除短信验证码错误")


    # 6. 将数据保存在数据库当中

    user = User()
    user.nick_name = mobile
    user.mobile = mobile

    # TODO 对密码进行加密 处理 不懂?

    user.password = password
    try:
        db.session.add(user)
        db.session.commit()
    except Exception as e:
        db.session.rollback()
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="数据库保存错误")

    # 7. 保存用户登录状态
    session["user_id"] = user.id
    session["nick_name"] = user.nick_name
    session["mobile"] = user.mobile

    # 返回登录成功
    return jsonify(errno=RET.OK, errmsg="OK")













    # # 1. 获取参数和判断是否有值
    # json_data = request.json
    # mobile = json_data.get("mobile")
    # sms_code = json_data.get("smscode")
    # password = json_data.get("password")
    #
    # if not all([mobile, sms_code, password]):
    #     # 参数不全
    #     return jsonify(errno=RET.PARAMERR, errmsg="参数不全")
    #
    # # 2. 从redis中获取指定手机号对应的短信验证码的
    # try:
    #     real_sms_code = redis_store.get("SMS_" + mobile)
    # except Exception as e:
    #     current_app.logger.error(e)
    #     # 获取本地验证码失败
    #     return jsonify(errno=RET.DBERR, errmsg="获取本地验证码失败")
    #
    # if not real_sms_code:
    #     # 短信验证码过期
    #     return jsonify(errno=RET.NODATA, errmsg="短信验证码过期")
    #
    # # 3. 校验验证码
    # if sms_code != real_sms_code:
    #     return jsonify(errno=RET.DATAERR, errmsg="短信验证码错误")
    # # 删除短信验证码
    # try:
    #     redis_store.delete("SMS_" + mobile)
    # except Exception as e:
    #     current_app.logger.error(e)
    #
    # # 4. 初始化 user 模型，并设置数据并添加到数据库
    # user = User()
    # user.nick_name = mobile
    # user.mobile = mobile
    # # 对密码进行处理
    # user.password = password
    #
    # try:
    #     db.session.add(user)
    #     db.session.commit()
    # except Exception as e:
    #     db.session.rollback()
    #     current_app.logger.error(e)
    #     # 数据保存错误
    #     return jsonify(errno=RET.DATAERR, errmsg="数据保存错误")
    # # 5. 保存用户登录状态
    # session["user_id"] = user.id
    # session["nick_name"] = user.nick_name
    # session["mobile"] = user.mobile
    #
    # # 6. 返回注册结果
    # return jsonify(errno=RET.OK, errmsg="OK")


@passport_blu.route('/login', methods=["POST"])
def login():
    """
    1. 获取参数和判断是否有值
    2. 从数据库查询出指定的用户
    3. 校验密码
    4. 保存用户登录状态
    5. 返回结果
    :return:
    """

    # 1. 获取参数和判断是否有值
    json_data = request.json

    mobile = json_data.get("mobile")
    password = json_data.get("password")

    if not all([mobile, password]):
        # 参数不全
        return jsonify(errno=RET.PARAMERR, errmsg="参数不全")

    # 2. 从数据库查询出指定的用户
    try:
        user = User.query.filter_by(mobile=mobile).first()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="查询数据错误")

    if not user:
        return jsonify(errno=RET.USERERR, errmsg="用户不存在")

    # 3. 校验密码
    if not user.check_passowrd(password):
        return jsonify(errno=RET.PWDERR, errmsg="密码错误")

    # 4. 保存用户登录状态
    session["user_id"] = user.id
    session["nick_name"] = user.nick_name
    session["mobile"] = user.mobile
    # 记录用户最后一次登录时间
    user.last_login = datetime.now()
    try:
        db.session.commit()
    except Exception as e:
        current_app.logger.error(e)
    # 5. 登录成功
    return jsonify(errno=RET.OK, errmsg="OK")


